Signal Quality Indicators for ECG signal
Copyright (C) 2022 Miodrag Bolic
This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details <https://www.gnu.org/licenses/>.
Acknowledgements
This notebook uses modified code written by Hooman Sedghamiz to compute fiducial points of the ECG signal. In addition ot uses modified ensemble averaging code from A practical guide to wave intensity analysis by Kim H. Parker. Dependencies: ecg.mat, MTEO_qrst.m, ens_averaging.m
% Changing the path from main_folder to a particular chapter
main_path=fileparts(which('Main_Content.mlx'));
%addpath(append(main_path,'/Chapter2'))
cd (append(main_path,'/Chapter4/Signal Quality'))
addpath(append(main_path,'/Service'))
SAVE_FLAG=0; % saving the figures in a file
Introduction
In this notebook, we are going to load ECG signal that is pretty clear and show the SQI based on 3 metrics:
- Kurtosis
- Spectral power ratio
- Template waveform
The signal is sampled at 2,500 samples per second and it is of high quality.
After that we will add Gaussian noise and impulse noise to the original signal and estimate the quality. Therefore, we will consider 3 cases:
Case 1: High quality ECG signal
Case 2: ECG signal with Gaussian noise and 60 Hz noise
Case 3: Impulse noise and high SNR
case123=3; % modify this number and run simulation for cases 1, 2 and 3
ecg = decimate(ecg,ratio);
noise=additive_noise_model(N, fs, 'Gaussian', [0, 0.1], '60 Hz', [0.1], 'Plotting');
noise=additive_noise_model(N, fs, 'BandLimited Impulse', [2, 1, 10], 'Gaussian', [0, 0.01],'Plotting');
title('Original signal with noise')
SQI
Kurtosis
Let us compute the kurtosis of the signal:
Spectral power ratio
% Define frequency ranges of interest fot the signal
Nfft = (60*2*NyquistF)/FResBPM; %number of bins in power spectrum
[Pxx,F] = periodogram(ecg,hamming(length(ecg)),Nfft,fs);
% Compute power in the frequency band of the signal of interest
FMask1 = (F >= Fmask_low_sig)&(F <= Fmask_high_sig);
PowerSig = sum(Pxx(FMask1));
% Compute power in the total frequency band
FMask2 = (F >= Fmask_low_all)&(F <= Fmask_high_all);
PowerAll = sum(Pxx(FMask2));
% Compute signal quality indices = ratio of the power in the
PowerRatio=PowerSig/PowerAll;
if PowerRatio < 0.9 && PowerRatio > 0.5
Template matching
Template matching includes:
- Filtering the signal
- Identifying pulses
- Extracting pulses and averaging them to obtain the template
- Calculating correlation between each pulse and the template
- Converting the correlation coefficient into a SQI score
%MTEO_qrst find all characteristic waves of ECG and performs low-pass filtering with 15 Hz cut-off frequency.
% R1(:,1) are time instances of R peaks the filtered ecg1 signal while
% R1(:,2) are th eamplitudes of the R peaks of the filtered signal ecg1.
[R1,Q,S,T_w,P_w, Te, ecg1] = MTEO_qrst(ecg,fs,0); % steps 1,2
% Ensample average of the non-filtered signal:
[pensavg, penssd, quality, r1]=ens_averaging(ecg, fs, R1(:,1), 1); % steps 3,4,5
title ('Non-filtered single beat waveforms and the template waveform')
DISPL=['No filtering, average corr coef=', num2str(mean (r1))];
disp(DISPL)
No filtering, average corr coef=0.91528
%plot(R1(1:end-1,1)*T,R1(1:end-1,2),'o')
text(R1(1:end-1,1)*T+0.3,R1(1:end-1,2),num2str(quality))
title('Noisy ECG signal with tSQI per beat')
% Ensample average of the filtered signal:
[pensavg, penssd, quality,r2]=ens_averaging(ecg1, fs, R1(:,1), 1);
title ('Filtered single beat waveforms and the template waveform')
annonation_save('b)',"Fig4.12b.jpg", SAVE_FLAG);
DISPL=['Bandpass filtering, average corr coef=', num2str(mean (r2))];
disp(DISPL)
Bandpass filtering, average corr coef=0.90861
plot(R1(1:end-1,1)*T,R1(1:end-1,2),'o')
text(R1(1:end-1,1)*T+0.3,R1(1:end-1,2),num2str(quality))
title('Filtered ECG signal with tSQI per beat')
annonation_save('a)',"Fig4.12a.jpg", SAVE_FLAG);
Exersizes
Excersize 1: repeat the analysis for cases 1, 2 and 3 and comment on the SQI indicators. Modify SNR in lines 14 and 18 to see the effect of noise on the signal quality indicators.
Excersize 2: Analyze effects of baseline drift on the quality indicators. Then filter out the baseline signal and observe the SQI again.